home *** CD-ROM | disk | FTP | other *** search
/ Internet Info 1994 March / Internet Info CD-ROM (Walnut Creek) (March 1994).iso / networking / ip / ka9q / src.arc / NRS.C < prev    next >
C/C++ Source or Header  |  1989-08-19  |  6KB  |  263 lines

  1. /* This module implements the serial line framing method used by
  2.  * net/rom nodes.  This allows the net/rom software to talk to
  3.  * an actual net/rom over its serial interface, which is useful
  4.  * if we want to do packet switching for multi-line wormholes.
  5.  * Dan Frank, W9NK
  6.  */
  7. #include <stdio.h>
  8. #include "global.h"
  9. #include "mbuf.h"
  10. #include "iface.h"
  11. #include "ax25.h"
  12. #include "nrs.h"
  13. #include "asy.h"
  14. #include "trace.h"
  15.  
  16. static struct mbuf *nrs_encode __ARGS((struct mbuf *bp));
  17. static struct mbuf *nrs_decode __ARGS((int16 dev,char c));
  18. static int nrsq __ARGS((int16 dev,struct mbuf *bp));
  19. static void nrasy_start __ARGS((int16 dev));
  20.  
  21. /* control structures, sort of overlayed on async control blocks */
  22. struct nrs Nrs[ASY_MAX];
  23.  
  24. /* Send a raw net/rom serial frame */
  25. int
  26. nrs_raw(iface,bp)
  27. struct iface *iface;
  28. struct mbuf *bp;
  29. {
  30.     dump(iface,IF_TRACE_OUT,TYPE_AX25,bp) ;
  31.     
  32.     /* Queue a frame on the output queue and start transmitter */
  33.     return nrsq(iface->dev,bp);
  34. }
  35.  
  36. /* Encode a raw packet in net/rom framing, put on link output queue, and kick
  37.  * transmitter
  38.  */
  39. static int
  40. nrsq(dev,bp)
  41. int16 dev;        /* Serial line number */
  42. struct mbuf *bp;    /* Buffer to be sent */
  43. {
  44.     register struct nrs *sp;
  45.  
  46.     if((bp = nrs_encode(bp)) == NULLBUF)
  47.         return -1;
  48.  
  49.     sp = &Nrs[dev];
  50.     enqueue(&sp->sndq,bp);
  51.     sp->sndcnt++;
  52.     if(sp->tbp == NULLBUF)
  53.         nrasy_start(dev);
  54.     return 0;
  55. }
  56.  
  57. /* Start output, if possible, on asynch device dev */
  58. static void
  59. nrasy_start(dev)
  60. int16 dev;
  61. {
  62.     register struct nrs *sp;
  63.  
  64.     if(!stxrdy(dev))
  65.         return;        /* Transmitter not ready */
  66.  
  67.     sp = &Nrs[dev];
  68.     if(sp->tbp != NULLBUF){
  69.         /* transmission just completed */
  70.         free_p(sp->tbp);
  71.         sp->tbp = NULLBUF;
  72.     }
  73.     if(sp->sndq == NULLBUF)
  74.         return;    /* No work */
  75.  
  76.     sp->tbp = dequeue(&sp->sndq);
  77.     sp->sndcnt--;
  78.     asy_output(dev,sp->tbp->data,sp->tbp->cnt);
  79. }
  80.  
  81. /* Encode a packet in net/rom serial format */
  82. static struct mbuf *
  83. nrs_encode(bp)
  84. struct mbuf *bp;
  85. {
  86.     struct mbuf *lbp;    /* Mbuf containing line-ready packet */
  87.     register char *cp;
  88.     char c;
  89.     unsigned char csum = 0 ;
  90.  
  91.     /* Allocate output mbuf that's twice as long as the packet.
  92.      * This is a worst-case guess (consider a packet full of STX's!)
  93.      * Add five bytes for STX, ETX, checksum, and two nulls.
  94.      */
  95.     lbp = alloc_mbuf(2*len_mbuf(bp) + 5);
  96.     if(lbp == NULLBUF){
  97.         /* No space; drop */
  98.         free_p(bp);
  99.         return NULLBUF;
  100.     }
  101.     cp = lbp->data;
  102.  
  103.     *cp++ = STX ;
  104.  
  105.     /* Copy input to output, escaping special characters */
  106.     while(pullup(&bp,&c,1) == 1){
  107.         switch(uchar(c)){
  108.         case STX:
  109.         case ETX:
  110.         case DLE:
  111.             *cp++ = DLE;
  112.             /* notice drop through to default */
  113.         default:
  114.             *cp++ = c;
  115.         }
  116.         csum += uchar(c) ;
  117.     }
  118.     *cp++ = ETX;
  119.     *cp++ = csum ;
  120.     *cp++ = NUL ;
  121.     *cp++ = NUL ;
  122.     
  123.     lbp->cnt = cp - lbp->data;
  124.     return lbp;
  125. }
  126. /* Process incoming bytes in net/rom serial format
  127.  * When a buffer is complete, return it; otherwise NULLBUF
  128.  */
  129. static struct mbuf *
  130. nrs_decode(dev,c)
  131. int16 dev;    /* net/rom unit number */
  132. char c;        /* Incoming character */
  133. {
  134.     struct mbuf *bp;
  135.     register struct nrs *sp;
  136.  
  137.     sp = &Nrs[dev];
  138.     switch(sp->state) {
  139.         case NRS_INTER:
  140.             if (uchar(c) == STX) {    /* look for start of frame */
  141.                 sp->state = NRS_INPACK ;    /* we're in a packet */
  142.                 sp->csum = 0 ;                /* reset checksum */
  143.             }
  144.             return NULLBUF ;
  145.         case NRS_CSUM:
  146.             bp = sp->rbp ;
  147.             sp->rbp = NULLBUF ;
  148.             sp->rcnt = 0 ;
  149.             sp->state = NRS_INTER ;    /* go back to inter-packet state */
  150.             if (sp->csum == uchar(c)) {
  151.                 sp->packets++ ;
  152.             } else {
  153.                 free_p(bp) ;    /* drop packet with bad checksum */
  154.                 bp = NULLBUF;
  155.                 sp->errors++ ;    /* increment error count */
  156.             }
  157.             return bp ;
  158.         case NRS_ESCAPE:
  159.             sp->state = NRS_INPACK ;    /* end of escape */
  160.             break ;            /* this will drop through to char processing */
  161.         case NRS_INPACK:
  162.             switch (uchar(c)) {
  163.             /* If we see an STX in a packet, assume that previous */
  164.             /* packet was trashed, and start a new packet */
  165.             case STX:
  166.                 free_p(sp->rbp) ;
  167.                 sp->rbp = NULLBUF ;
  168.                 sp->rcnt = 0 ;
  169.                 sp->csum = 0 ;
  170.                 sp->errors++ ;
  171.                 return NULLBUF ;
  172.             case ETX:
  173.                 sp->state = NRS_CSUM ;    /* look for checksum */
  174.                 return NULLBUF ;
  175.             case DLE:
  176.                 sp->state = NRS_ESCAPE ;
  177.                 return NULLBUF ;
  178.             }
  179.     }
  180.     /* If we get to here, it's with a character that's part of the packet.
  181.      * Make sure there's space for it.
  182.      */
  183.     if(sp->rbp == NULLBUF){
  184.         /* Allocate first mbuf for new packet */
  185.         if((sp->rbp1 = sp->rbp = alloc_mbuf(NRS_ALLOC)) == NULLBUF) {
  186.             sp->state = NRS_INTER ;
  187.             return NULLBUF; /* No memory, drop */
  188.         }
  189.         sp->rcp = sp->rbp->data;
  190.     } else if(sp->rbp1->cnt == NRS_ALLOC){
  191.         /* Current mbuf is full; link in another */
  192.         if((sp->rbp1->next = alloc_mbuf(NRS_ALLOC)) == NULLBUF){
  193.             /* No memory, drop whole thing */
  194.             free_p(sp->rbp);
  195.             sp->rbp = NULLBUF;
  196.             sp->rcnt = 0;
  197.             sp->state = NRS_INTER ;
  198.             return NULLBUF;
  199.         }
  200.         sp->rbp1 = sp->rbp1->next;
  201.         sp->rcp = sp->rbp1->data;
  202.     }
  203.     /* Store the character, increment fragment and total
  204.      * byte counts
  205.      */
  206.     *sp->rcp++ = c;
  207.     sp->rbp1->cnt++;
  208.     sp->rcnt++;
  209.     sp->csum += uchar(c) ;    /* add to checksum */
  210.     return NULLBUF;
  211. }
  212.  
  213. /* Process net/rom serial line I/O */
  214. void
  215. nrs_recv(dev,v1,v2)
  216. int dev;
  217. void *v1;
  218. void *v2;
  219. {
  220.     char c;
  221.     struct mbuf *bp,*nbp;
  222.     struct phdr *phdr;
  223.  
  224.     /* Process any pending input */
  225.     for(;;){
  226.         c = get_asy(dev);
  227.         if((bp = nrs_decode(dev,c)) == NULLBUF)
  228.             continue;
  229.         if((nbp = pushdown(bp,sizeof(struct phdr))) == NULLBUF){
  230.             free_p(bp);
  231.             continue;
  232.         }
  233.         phdr = (struct phdr *)nbp->data;
  234.         phdr->iface = Nrs[dev].iface;
  235.         phdr->type = TYPE_AX25;
  236.         enqueue(&Hopper,nbp);
  237.         /* Kick the transmitter if it's idle */
  238.         if(stxrdy(dev))
  239.             nrasy_start(dev);
  240.     }
  241.  
  242. }
  243. /* donrstat:  display status of active net/rom serial interfaces */
  244. int
  245. donrstat(argc,argv,p)
  246. int argc ;
  247. char *argv[] ;
  248. void *p;
  249. {
  250.     register struct nrs *np ;
  251.     register int i ;
  252.  
  253.     printf("Interface  SndQ  RcvB  NumReceived  CSumErrors\n") ;
  254.  
  255.     for (i = 0, np = Nrs ; i < ASY_MAX ; i++, np++)
  256.         if (np->iface != NULLIF)
  257.             printf(" %8s   %3d  %4d   %10lu  %10lu\n",
  258.                     np->iface->name, np->sndcnt, np->rcnt,
  259.                     np->packets, np->errors) ;
  260.  
  261.     return 0 ;
  262. }
  263.